<template><div><h2 id="扩展-extensions" tabindex="-1"><a class="header-anchor" href="#扩展-extensions"><span>扩展（Extensions）</span></a></h2>
<p>扩展是专门设计的在 Yii 应用中随时可拿来使用的，并可重发布的软件包。例如，<a href="https://github.com/yiisoft/yii2-debug" target="_blank" rel="noopener noreferrer">yiisoft/yii2-debug</a> 扩展在你的应用的每个页面底部添加一个方便用于调试的工具栏，帮助你简单地抓取页面生成的情况。你可以使用扩展来加速你的开发过程。</p>
<blockquote>
<p>Info: 本文中我们使用的术语 “扩展” 特指 Yii 软件包。而用术语 “软件包” 和 “库” 指代非 Yii 专用的通常意义上的软件包。</p>
</blockquote>
<h2 id="使用扩展" tabindex="-1"><a class="header-anchor" href="#使用扩展"><span>使用扩展</span></a></h2>
<p>要使用扩展，你要先安装它。大多数扩展以 <a href="https://getcomposer.org/" target="_blank" rel="noopener noreferrer">Composer</a> 软件包的形式发布，这样的扩展可采取下述两个步骤来安装：</p>
<ol>
<li>修改你的应用的 <code v-pre>composer.json</code> 文件，指明你要安装的是哪个扩展（Composer 软件包）；</li>
<li>运行 <code v-pre>composer install</code> 来安装指定的扩展。</li>
</ol>
<p>注意如果你还没有安装 <a href="https://getcomposer.org/" target="_blank" rel="noopener noreferrer">Composer</a> ，你需要先安装。</p>
<p>默认情况，Composer 安装的是在 <a href="https://packagist.org/" target="_blank" rel="noopener noreferrer">Packagist</a> 中注册的软件包，即最大的开源 Composer 代码库。你可以在 Packageist 中查找扩展。你也可以 <a href="https://getcomposer.org/doc/05-repositories.md#repository" target="_blank" rel="noopener noreferrer">创建你自己的代码库</a> 然后配置 Composer 来使用它。如果是在开发私有的扩展，并且想只在你的其他工程中共享时，这样做是很有用的。</p>
<p>通过 Composer 安装的扩展会存放在 <code v-pre>BasePath/vendor</code> 目录下，这里的 <code v-pre>BasePath</code> 指你的应用的 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/structure-applications.md#basePath" target="_blank" rel="noopener noreferrer">base path</a>。因为 Composer 还是一个依赖管理器，当它安装一个包时，也将安装这个包所依赖的所有软件包。</p>
<p>例如想安装 <code v-pre>yiisoft/yii2-imagine</code> 扩展，可按如下示例修改你的 <code v-pre>composer.json</code> 文件：</p>
<div class="language-json line-numbers-mode" data-highlighter="prismjs" data-ext="json" data-title="json"><pre v-pre class="language-json"><code><span class="line"><span class="token punctuation">{</span></span>
<span class="line">    <span class="token comment">// ...</span></span>
<span class="line"></span>
<span class="line">    <span class="token property">"require"</span><span class="token operator">:</span> <span class="token punctuation">{</span></span>
<span class="line">        <span class="token comment">// ... other dependencies</span></span>
<span class="line"></span>
<span class="line">        <span class="token property">"yiisoft/yii2-imagine"</span><span class="token operator">:</span> <span class="token string">"~2.0.0"</span></span>
<span class="line">    <span class="token punctuation">}</span></span>
<span class="line"><span class="token punctuation">}</span></span>
<span class="line"></span></code></pre>
<div class="line-numbers" aria-hidden="true" style="counter-reset:line-number 0"><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div></div></div><p>安装完成后，你应该能在 <code v-pre>BasePath/vendor</code> 目录下见到 <code v-pre>yiisoft/yii2-imagine</code> 目录。你也应该 见到另一个 <code v-pre>imagine/imagine</code> 目录，在其中安装了所依赖的包。</p>
<blockquote>
<p>Info: <code v-pre>yiisoft/yii2-imagine</code> 是 Yii 由开发团队维护一个核心扩展，所有核心扩展均由 <a href="https://packagist.org/" target="_blank" rel="noopener noreferrer">Packagist</a> 集中管理，命名为 <code v-pre>yiisoft/yii2-xyz</code>，其中的 <code v-pre>xyz</code>，不同扩展有不同名称。</p>
</blockquote>
<p>现在你可以使用安装好的扩展了，好比是应用的一部分。如下示例展示了如何使用 <code v-pre>yiisoft/yii2-imagine</code> 扩展提供的 <code v-pre>yii\imagine\Image</code> 类：</p>
<div class="language-php line-numbers-mode" data-highlighter="prismjs" data-ext="php" data-title="php"><pre v-pre class="language-php"><code><span class="line"><span class="token keyword">use</span> <span class="token package">Yii</span><span class="token punctuation">;</span></span>
<span class="line"><span class="token keyword">use</span> <span class="token package">yii<span class="token punctuation">\</span>imagine<span class="token punctuation">\</span>Image</span><span class="token punctuation">;</span></span>
<span class="line"></span>
<span class="line"><span class="token comment">// generate a thumbnail image</span></span>
<span class="line"><span class="token class-name static-context">Image</span><span class="token operator">::</span><span class="token function">thumbnail</span><span class="token punctuation">(</span><span class="token string single-quoted-string">'@webroot/img/test-image.jpg'</span><span class="token punctuation">,</span> <span class="token number">120</span><span class="token punctuation">,</span> <span class="token number">120</span><span class="token punctuation">)</span></span>
<span class="line">    <span class="token operator">-></span><span class="token function">save</span><span class="token punctuation">(</span><span class="token class-name static-context">Yii</span><span class="token operator">::</span><span class="token function">getAlias</span><span class="token punctuation">(</span><span class="token string single-quoted-string">'@runtime/thumb-test-image.jpg'</span><span class="token punctuation">)</span><span class="token punctuation">,</span> <span class="token punctuation">[</span><span class="token string single-quoted-string">'quality'</span> <span class="token operator">=></span> <span class="token number">50</span><span class="token punctuation">]</span><span class="token punctuation">)</span><span class="token punctuation">;</span></span>
<span class="line"></span></code></pre>
<div class="line-numbers" aria-hidden="true" style="counter-reset:line-number 0"><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div></div></div><blockquote>
<p>Info: 扩展类由 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/concept-autoloading.md" target="_blank" rel="noopener noreferrer">Yii class autoloader</a> 自动加载。</p>
</blockquote>
<h3 id="手动安装扩展" tabindex="-1"><a class="header-anchor" href="#手动安装扩展"><span>手动安装扩展</span></a></h3>
<p>在极少情况下，你可能需要手动安装一部分或者全部扩展，而不是依赖 Composer。想做到这一点，你应当：</p>
<ol>
<li>下载扩展压缩文件，解压到 <code v-pre>vendor</code> 目录。</li>
<li>如果有，则安装扩展提供的自动加载器。</li>
<li>按指导说明下载和安装所有依赖的扩展。</li>
</ol>
<p>如果扩展没有提供类的自动加载器，但也遵循了 <a href="http://www.php-fig.org/psr/psr-4/" target="_blank" rel="noopener noreferrer">PSR-4 standard</a> 标准，那么你可以使用 Yii 提供的类自动加载器来加载扩展类。你需要做的仅仅是为扩展的根目录声明一个 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/concept-aliases.md#defining-aliases" target="_blank" rel="noopener noreferrer">root alias</a>。例如，假设在 <code v-pre>vendor/mycompany/myext</code> 目录中安装了一个扩展，并且扩展类的命名空间为 <code v-pre>myext</code> ，那么你可以在应用配置文件中包含如下代码：</p>
<div class="language-php line-numbers-mode" data-highlighter="prismjs" data-ext="php" data-title="php"><pre v-pre class="language-php"><code><span class="line"><span class="token punctuation">[</span></span>
<span class="line">    <span class="token string single-quoted-string">'aliases'</span> <span class="token operator">=></span> <span class="token punctuation">[</span></span>
<span class="line">        <span class="token string single-quoted-string">'@myext'</span> <span class="token operator">=></span> <span class="token string single-quoted-string">'@vendor/mycompany/myext'</span><span class="token punctuation">,</span></span>
<span class="line">    <span class="token punctuation">]</span><span class="token punctuation">,</span></span>
<span class="line"><span class="token punctuation">]</span></span>
<span class="line"></span></code></pre>
<div class="line-numbers" aria-hidden="true" style="counter-reset:line-number 0"><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div></div></div><h2 id="创建扩展" tabindex="-1"><a class="header-anchor" href="#创建扩展"><span>创建扩展</span></a></h2>
<p>在你需要将你的杰作分享给其他人的时候，你可能会考虑创建一个扩展。扩展可包括任何你喜欢的代码，例如助手类、小部件、模块，等等。</p>
<p>建议你按照 <a href="https://getcomposer.org/" target="_blank" rel="noopener noreferrer">Composer package</a> 的条款创建扩展，以便其他人更容易安装和使用。就像上面的章节讲述的那样。</p>
<p>以下是将扩展创建为一个 Composer 软件包的需遵循的基本步骤。</p>
<ol>
<li>为你的扩展建一个工程，并将它存放在版本控制代码库中，例如 <a href="https://github.com/" target="_blank" rel="noopener noreferrer">github.com</a> 。扩展的开发和维护都应该在这个代码库中进行。</li>
<li>在工程的根目录下，建一个 Composer 所需的名为 <code v-pre>composer.json</code> 的文件。详情请参考后面的章节。</li>
<li>在一个 Composer 代码库中注册你的扩展，比如在 <a href="https://packagist.org/" target="_blank" rel="noopener noreferrer">Packagist</a> 中，以便其他用户能找到以及用 Composer 安装你的扩展。</li>
</ol>
<h3 id="composer-json" tabindex="-1"><a class="header-anchor" href="#composer-json"><span><code v-pre>composer.json</code></span></a></h3>
<p>每个 Composer 软件包在根目录都必须有一个 <code v-pre>composer.json</code> 文件。该文件包含软件包的元数据。你可以在 <a href="https://getcomposer.org/doc/01-basic-usage.md#composer-json-project-setup" target="_blank" rel="noopener noreferrer">Composer手册</a> 中找到完整关于该文件的规格。以下例子展示了 <code v-pre>yiisoft/yii2-imagine</code> 扩展的 <code v-pre>composer.json</code> 文件。</p>
<div class="language-json line-numbers-mode" data-highlighter="prismjs" data-ext="json" data-title="json"><pre v-pre class="language-json"><code><span class="line"><span class="token punctuation">{</span></span>
<span class="line">    <span class="token comment">// package name</span></span>
<span class="line">    <span class="token property">"name"</span><span class="token operator">:</span> <span class="token string">"yiisoft/yii2-imagine"</span><span class="token punctuation">,</span></span>
<span class="line"></span>
<span class="line">    <span class="token comment">// package type</span></span>
<span class="line">    <span class="token property">"type"</span><span class="token operator">:</span> <span class="token string">"yii2-extension"</span><span class="token punctuation">,</span></span>
<span class="line"></span>
<span class="line">    <span class="token property">"description"</span><span class="token operator">:</span> <span class="token string">"The Imagine integration for the Yii framework"</span><span class="token punctuation">,</span></span>
<span class="line">    <span class="token property">"keywords"</span><span class="token operator">:</span> <span class="token punctuation">[</span><span class="token string">"yii2"</span><span class="token punctuation">,</span> <span class="token string">"imagine"</span><span class="token punctuation">,</span> <span class="token string">"image"</span><span class="token punctuation">,</span> <span class="token string">"helper"</span><span class="token punctuation">]</span><span class="token punctuation">,</span></span>
<span class="line">    <span class="token property">"license"</span><span class="token operator">:</span> <span class="token string">"BSD-3-Clause"</span><span class="token punctuation">,</span></span>
<span class="line">    <span class="token property">"support"</span><span class="token operator">:</span> <span class="token punctuation">{</span></span>
<span class="line">        <span class="token property">"issues"</span><span class="token operator">:</span> <span class="token string">"https://github.com/yiisoft/yii2/issues?labels=ext%3Aimagine"</span><span class="token punctuation">,</span></span>
<span class="line">        <span class="token property">"forum"</span><span class="token operator">:</span> <span class="token string">"https://forum.yiiframework.com/"</span><span class="token punctuation">,</span></span>
<span class="line">        <span class="token property">"wiki"</span><span class="token operator">:</span> <span class="token string">"https://www.yiiframework.com/wiki/"</span><span class="token punctuation">,</span></span>
<span class="line">        <span class="token property">"irc"</span><span class="token operator">:</span> <span class="token string">"ircs://irc.libera.chat:6697/yii"</span><span class="token punctuation">,</span></span>
<span class="line">        <span class="token property">"source"</span><span class="token operator">:</span> <span class="token string">"https://github.com/yiisoft/yii2"</span></span>
<span class="line">    <span class="token punctuation">}</span><span class="token punctuation">,</span></span>
<span class="line">    <span class="token property">"authors"</span><span class="token operator">:</span> <span class="token punctuation">[</span></span>
<span class="line">        <span class="token punctuation">{</span></span>
<span class="line">            <span class="token property">"name"</span><span class="token operator">:</span> <span class="token string">"Antonio Ramirez"</span><span class="token punctuation">,</span></span>
<span class="line">            <span class="token property">"email"</span><span class="token operator">:</span> <span class="token string">"amigo.cobos@gmail.com"</span></span>
<span class="line">        <span class="token punctuation">}</span></span>
<span class="line">    <span class="token punctuation">]</span><span class="token punctuation">,</span></span>
<span class="line"></span>
<span class="line">    <span class="token comment">// package dependencies</span></span>
<span class="line">    <span class="token property">"require"</span><span class="token operator">:</span> <span class="token punctuation">{</span></span>
<span class="line">        <span class="token property">"yiisoft/yii2"</span><span class="token operator">:</span> <span class="token string">"~2.0.0"</span><span class="token punctuation">,</span></span>
<span class="line">        <span class="token property">"imagine/imagine"</span><span class="token operator">:</span> <span class="token string">"v0.5.0"</span></span>
<span class="line">    <span class="token punctuation">}</span><span class="token punctuation">,</span></span>
<span class="line"></span>
<span class="line">    <span class="token comment">// class autoloading specs</span></span>
<span class="line">    <span class="token property">"autoload"</span><span class="token operator">:</span> <span class="token punctuation">{</span></span>
<span class="line">        <span class="token property">"psr-4"</span><span class="token operator">:</span> <span class="token punctuation">{</span></span>
<span class="line">            <span class="token property">"yii\\imagine\\"</span><span class="token operator">:</span> <span class="token string">""</span></span>
<span class="line">        <span class="token punctuation">}</span></span>
<span class="line">    <span class="token punctuation">}</span></span>
<span class="line"><span class="token punctuation">}</span></span>
<span class="line"></span></code></pre>
<div class="line-numbers" aria-hidden="true" style="counter-reset:line-number 0"><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div></div></div><h4 id="包名" tabindex="-1"><a class="header-anchor" href="#包名"><span>包名</span></a></h4>
<p>每个 Composer 软件包都应当有一个唯一的包名以便能从其他的软件包中识别出来。包名的格式为 <code v-pre>vendorName/projectName</code> 。例如在包名 <code v-pre>yiisoft/yii2-imagine</code> 中，vendor 名和 project 名分别是 <code v-pre>yiisoft</code> 和 <code v-pre>yii2-imagine</code> 。</p>
<p>不要用 <code v-pre>yiisoft</code> 作为你的 vendor 名，由于它被 Yii 的核心代码预留使用了。</p>
<p>我们推荐你用 <code v-pre>yii2-</code> 作为你的包名的前缀，表示它是 Yii 2 的扩展，例如，<code v-pre>myname/yii2-mywidget</code>。这更便于用户辨别是否是 Yii 2 的扩展。</p>
<h4 id="包类型" tabindex="-1"><a class="header-anchor" href="#包类型"><span>包类型</span></a></h4>
<p>将你的扩展指明为 <code v-pre>yii2-extension</code> 类型很重要，以便安装的时候 能被识别出是一个 Yii 扩展。</p>
<p>当用户运行 <code v-pre>composer install</code> 安装一个扩展时，<code v-pre>vendor/yiisoft/extensions.php</code> 文件会被自动更新使之包含新扩展的信息。从该文件中，Yii 应用程序就能知道安装了哪些扩展 (这些信息可通过 [[yii\base\Application::extensions]] 访问)。</p>
<h4 id="依赖" tabindex="-1"><a class="header-anchor" href="#依赖"><span>依赖</span></a></h4>
<p>你的扩展依赖于 Yii（理所当然）。因此你应当在 <code v-pre>composer.json</code> 文件中列出它 (<code v-pre>yiisoft/yii2</code>)。如果你的扩展还依赖其他的扩展或者是第三方库，你也要一并列出来。确定你也为每一个依赖的包列出了适当的版本约束条件 (比如 <code v-pre>1.*</code>, <code v-pre>@stable</code>) 。当你发布一个稳定版本时，你所依赖的包也应当使用稳定版本。</p>
<p>大多数 JS/CSS 包是用 <a href="http://bower.io/" target="_blank" rel="noopener noreferrer">Bower</a> 来管理的，而非 Composer。你可使用 <a href="https://github.com/fxpio/composer-asset-plugin" target="_blank" rel="noopener noreferrer">Composer asset 插件</a> 使之可以 通过 Composer 来管理这类包。如果你的扩展依赖 Bower 软件包，你可以如下例所示那样简单地 在 <code v-pre>composer.json</code> 文件的依赖中列出它。</p>
<div class="language-json line-numbers-mode" data-highlighter="prismjs" data-ext="json" data-title="json"><pre v-pre class="language-json"><code><span class="line"><span class="token punctuation">{</span></span>
<span class="line">    <span class="token comment">// package dependencies</span></span>
<span class="line">    <span class="token property">"require"</span><span class="token operator">:</span> <span class="token punctuation">{</span></span>
<span class="line">        <span class="token property">"bower-asset/jquery"</span><span class="token operator">:</span> <span class="token string">">=1.11.*"</span></span>
<span class="line">    <span class="token punctuation">}</span></span>
<span class="line"><span class="token punctuation">}</span></span>
<span class="line"></span></code></pre>
<div class="line-numbers" aria-hidden="true" style="counter-reset:line-number 0"><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div></div></div><p>上述代码表明该扩展依赖于 <code v-pre>jquery</code> Bower 包。一般来说，你可以在 <code v-pre>composer.json</code> 中用 <code v-pre>bower-asset/PackageName</code> 指定 Bower 包，用 <code v-pre>npm-asset/PackageName</code> 指定 NPM 包。 当 Compower 安装 Bower 和 NPM 软件包时，包的内容默认会分别安装到 <code v-pre>@vendor/bower/PackageName</code> 和 <code v-pre>@vendor/npm/Packages</code> 下。这两个目录还可以分别用 <code v-pre>@bower/PackageName</code> 和 <code v-pre>@npm/PackageName</code> 别名指向。</p>
<p>关于 asset 管理的详细情况，请参照 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/structure-assets.md#bower-npm-assets" target="_blank" rel="noopener noreferrer">前端资源</a> 章节。</p>
<h4 id="类的自动加载" tabindex="-1"><a class="header-anchor" href="#类的自动加载"><span>类的自动加载</span></a></h4>
<p>为使你的类能够被 Yii 的类自动加载器或者 Composer 的类自动加载器自动加载，你应当在 <code v-pre>composer.json</code> 中指定 <code v-pre>autoload</code> 条目，如下所示：</p>
<div class="language-json line-numbers-mode" data-highlighter="prismjs" data-ext="json" data-title="json"><pre v-pre class="language-json"><code><span class="line"><span class="token punctuation">{</span></span>
<span class="line">    <span class="token comment">// ....</span></span>
<span class="line"></span>
<span class="line">    <span class="token property">"autoload"</span><span class="token operator">:</span> <span class="token punctuation">{</span></span>
<span class="line">        <span class="token property">"psr-4"</span><span class="token operator">:</span> <span class="token punctuation">{</span></span>
<span class="line">            <span class="token property">"yii\\imagine\\"</span><span class="token operator">:</span> <span class="token string">""</span></span>
<span class="line">        <span class="token punctuation">}</span></span>
<span class="line">    <span class="token punctuation">}</span></span>
<span class="line"><span class="token punctuation">}</span></span>
<span class="line"></span></code></pre>
<div class="line-numbers" aria-hidden="true" style="counter-reset:line-number 0"><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div></div></div><p>你可以列出一个或者多个根命名空间和它们的文件目录。</p>
<p>当扩展安装到应用中后，Yii 将为每个所列出根命名空间创建一个 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/concept-aliases.md#extension-aliases" target="_blank" rel="noopener noreferrer">别名</a> 指向命名空间对应的目录。例如，上述的 <code v-pre>autoload</code> 条目声明将对应于别名 <code v-pre>@yii/imagine</code>。</p>
<h3 id="推荐的做法" tabindex="-1"><a class="header-anchor" href="#推荐的做法"><span>推荐的做法</span></a></h3>
<p>扩展意味着会被其他人使用，你在开发中通常需要额外的付出。下面我们介绍一些通用的及推荐的做法，以创建高品质的扩展。</p>
<h4 id="命名空间" tabindex="-1"><a class="header-anchor" href="#命名空间"><span>命名空间</span></a></h4>
<p>为避免冲突以及使你的扩展中的类能被自动加载，你的类应当使用命名空间，并使类的命名符合 <a href="http://www.php-fig.org/psr/psr-4/" target="_blank" rel="noopener noreferrer">PSR-4 standard</a> 或者 <a href="http://www.php-fig.org/psr/psr-0/" target="_blank" rel="noopener noreferrer">PSR-0 standard</a> 标准。</p>
<p>你的类的命名空间应以 <code v-pre>vendorName\extensionName</code> 起始，其中 <code v-pre>extensionName</code> 和项目名相同，除了它没有 <code v-pre>yii2-</code> 前缀外。例如，对 <code v-pre>yiisoft/yii2-imagine</code> 扩展 来说，我们用 <code v-pre>yii\imagine</code> 作为它的类的命名空间。</p>
<p>不要使用 <code v-pre>yii</code>、<code v-pre>yii2</code> 或者 <code v-pre>yiisoft</code> 作为你的 vendor 名。这些名称已由 Yii 内核代码预留使用了。</p>
<h4 id="类的自举引导" tabindex="-1"><a class="header-anchor" href="#类的自举引导"><span>类的自举引导</span></a></h4>
<p>有时候，你可能想让你的扩展在应用的 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/runtime-bootstrapping.md" target="_blank" rel="noopener noreferrer">自举过程</a> 中执行一些代码。例如，你的扩展可能想响应应用的 <code v-pre>beginRequest</code> 事件，做一些环境的设置工作。虽然你可以指导扩展的使用者显式地将你的扩展中的事件句柄附加（绑定）到 <code v-pre>beginRequest</code> 事件，但是更好的方法是自动完成。</p>
<p>为实现该目标，你可以创建一个所谓 <em>bootstrapping class</em>（自举类）实现 [[yii\base\BootstrapInterface]] 接口。例如，</p>
<div class="language-php line-numbers-mode" data-highlighter="prismjs" data-ext="php" data-title="php"><pre v-pre class="language-php"><code><span class="line"><span class="token keyword">namespace</span> <span class="token package">myname<span class="token punctuation">\</span>mywidget</span><span class="token punctuation">;</span></span>
<span class="line"></span>
<span class="line"><span class="token keyword">use</span> <span class="token package">yii<span class="token punctuation">\</span>base<span class="token punctuation">\</span>BootstrapInterface</span><span class="token punctuation">;</span></span>
<span class="line"><span class="token keyword">use</span> <span class="token package">yii<span class="token punctuation">\</span>base<span class="token punctuation">\</span>Application</span><span class="token punctuation">;</span></span>
<span class="line"></span>
<span class="line"><span class="token keyword">class</span> <span class="token class-name-definition class-name">MyBootstrapClass</span> <span class="token keyword">implements</span> <span class="token class-name">BootstrapInterface</span></span>
<span class="line"><span class="token punctuation">{</span></span>
<span class="line">    <span class="token keyword">public</span> <span class="token keyword">function</span> <span class="token function-definition function">bootstrap</span><span class="token punctuation">(</span><span class="token variable">$app</span><span class="token punctuation">)</span></span>
<span class="line">    <span class="token punctuation">{</span></span>
<span class="line">        <span class="token variable">$app</span><span class="token operator">-></span><span class="token function">on</span><span class="token punctuation">(</span><span class="token class-name static-context">Application</span><span class="token operator">::</span><span class="token constant">EVENT_BEFORE_REQUEST</span><span class="token punctuation">,</span> <span class="token keyword">function</span> <span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span></span>
<span class="line">             <span class="token comment">// do something here</span></span>
<span class="line">        <span class="token punctuation">}</span><span class="token punctuation">)</span><span class="token punctuation">;</span></span>
<span class="line">    <span class="token punctuation">}</span></span>
<span class="line"><span class="token punctuation">}</span></span>
<span class="line"></span></code></pre>
<div class="line-numbers" aria-hidden="true" style="counter-reset:line-number 0"><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div></div></div><p>然后你将这个类在 <code v-pre>composer.json</code> 文件中列出来，如下所示，</p>
<div class="language-json line-numbers-mode" data-highlighter="prismjs" data-ext="json" data-title="json"><pre v-pre class="language-json"><code><span class="line"><span class="token punctuation">{</span></span>
<span class="line">    <span class="token comment">// ...</span></span>
<span class="line"></span>
<span class="line">    <span class="token property">"extra"</span><span class="token operator">:</span> <span class="token punctuation">{</span></span>
<span class="line">        <span class="token property">"bootstrap"</span><span class="token operator">:</span> <span class="token string">"myname\\mywidget\\MyBootstrapClass"</span></span>
<span class="line">    <span class="token punctuation">}</span></span>
<span class="line"><span class="token punctuation">}</span></span>
<span class="line"></span></code></pre>
<div class="line-numbers" aria-hidden="true" style="counter-reset:line-number 0"><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div><div class="line-number"></div></div></div><p>当这个扩展安装到应用后，Yii 将在每一个请求的自举过程中自动实例化自举类并调用其 [[yii\base\BootstrapInterface::bootstrap()]] 方法。</p>
<h4 id="操作数据库" tabindex="-1"><a class="header-anchor" href="#操作数据库"><span>操作数据库</span></a></h4>
<p>你的扩展可能要存取数据库。不要假设使用你的扩展的应用总是用 <code v-pre>Yii::$db</code> 作为数据库连接。你应当在需要访问数据库的类中申明一个 <code v-pre>db</code> 属性。这个属性允许你的扩展的用户可定制你的扩展使用哪个 DB 连接。例如，你可以参考 [[yii\caching\DbCache]] 类看一下它是如何申明和使用 <code v-pre>db</code> 属性的。</p>
<p>如果你的扩展需要创建特定的数据库表，或者修改数据库结构，你应当</p>
<ul>
<li>提供 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/db-migrations.md" target="_blank" rel="noopener noreferrer">数据迁移</a> 来操作数据库的结构修改，而不是使用SQL文本文件；</li>
<li>尽量使迁移文件适用于不同的 DBMS；</li>
<li>在迁移文件中避免使用 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/db-active-record.md" target="_blank" rel="noopener noreferrer">Active Record</a>。</li>
</ul>
<h4 id="使用-assets" tabindex="-1"><a class="header-anchor" href="#使用-assets"><span>使用 Assets</span></a></h4>
<p>如果你的扩展是挂件或者模块类型，它有可能需要使用一些 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/structure-assets.md" target="_blank" rel="noopener noreferrer">assets</a> 。 例如，一个模块可能要显示一些包含图片，JavaScript 和 CSS 的页面。因为扩展的文件 都是放在同一个目录之下，安装之后 Web 无法读取，你有两个选择使得这些 asset 文件目录 可以通过 Web 读取：</p>
<ul>
<li>让扩展的用户手动将这些 asset 文件拷贝到特定的 Web 可以读取的文件夹；</li>
<li>申明一个 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/structure-assets.md" target="_blank" rel="noopener noreferrer">asset bundle</a> 并依靠 asset 发布机制自动将这些文件（asset bundle 中列出的文件） 拷贝到 Web 可读的文件夹。</li>
</ul>
<p>我们推荐你使用第二种方法，以便其他人能更容易使用你的扩展。 更详细的关于如何处理 assets ，请参照 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/structure-assets.md" target="_blank" rel="noopener noreferrer">Assets</a> 章节。</p>
<h4 id="国际化和本地化" tabindex="-1"><a class="header-anchor" href="#国际化和本地化"><span>国际化和本地化</span></a></h4>
<p>你的扩展可能会在支持不同语言的应用中使用！因此，如果你的扩展要显示内容给终端用户， 你应当试着实现 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/tutorial-i18n.md" target="_blank" rel="noopener noreferrer">国际化和本地化</a>，特别地，</p>
<ul>
<li>如果扩展为终端用户显示信息，这些信息应该用 <code v-pre>Yii::t()</code> 包装起来，以便可以进行翻译。 只给开发者参考的信息（如内部异常信息）不需要做翻译。</li>
<li>如果扩展显示数字、日期等，你应该用 [[yii\i18n\Formatter]] 中适当的格式化规则做格式化处理。</li>
</ul>
<p>更详细的信息，请参照 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/tutorial-i18n.md" target="_blank" rel="noopener noreferrer">Internationalization</a> 章节。</p>
<h4 id="测试" tabindex="-1"><a class="header-anchor" href="#测试"><span>测试</span></a></h4>
<p>你一定想让你的扩展可以无暇地运行而不会给其他人带来问题和麻烦。为达到这个目的，你应当在公开发布前做测试。</p>
<p>推荐你创建测试用例，做全面覆盖的测试你的扩展，而不只是依赖于手动测试。每次发布新版本前，你只要简单地运行这些测试用例确保一切完好。Yii 提供了测试支持，使你更容易写单元测试、验收测试和功能测试。详情请参照 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/test-overview.md" target="_blank" rel="noopener noreferrer">测试</a> 章节。</p>
<h4 id="版本控制" tabindex="-1"><a class="header-anchor" href="#版本控制"><span>版本控制</span></a></h4>
<p>你应该为每一个扩展定一个版本号（如 <code v-pre>1.0.1</code>）。我们推荐你命名版本号时参照 <a href="http://semver.org/" target="_blank" rel="noopener noreferrer">semantic versioning</a> 决定用什么样的版本号。</p>
<h4 id="发布" tabindex="-1"><a class="header-anchor" href="#发布"><span>发布</span></a></h4>
<p>为使其他人知道你的扩展，你应该公开发布。</p>
<p>如果你首次发布一个扩展，你应该在 Composer 代码库中注册它，例如 <a href="https://packagist.org/" target="_blank" rel="noopener noreferrer">Packagist</a>。之后，你所需要做的仅仅是在 版本管理库中创建一个 tag （如<code v-pre>v1.0.1</code>），然后通知 Composer 代码库。其他人就能查找到这个新的发布了，并可通过 Composer 代码库安装和更新该扩展。</p>
<p>在发布你的扩展时，除了代码文件，你还应该考虑包含如下内容 帮助其他人了解和使用你的扩展：</p>
<ul>
<li>根目录下的 readme 文件：它描述你的扩展是干什么的以及如何安装和使用。我们推荐你用 <a href="http://daringfireball.net/projects/markdown/" target="_blank" rel="noopener noreferrer">Markdown</a> 的格式来写并将文件命名为 <code v-pre>readme.md</code>。</li>
<li>根目录下的修改日志文件：它列举每个版本的发布做了哪些更改。该文件可以用 Markdown 根式 编写并命名为 <code v-pre>changelog.md</code>。</li>
<li>根目录下的升级文件：它给出如何从其他就版本升级该扩展的指导。该文件可以用 Markdown 根式 编写并命名为 <code v-pre>changelog.md</code>。</li>
<li>入门指南、演示代码、截屏图示等：如果你的扩展提供了许多功能，在 readme 文件中不能完整 描述时，就要用到这些文件。</li>
<li>API 文档：你的代码应当做好文档，让其他人更容易阅读和理解。你可以参照 <a href="https://github.com/yiisoft/yii2/blob/master/framework/base/Object.php" target="_blank" rel="noopener noreferrer">Object class file</a> 学习如何为你的代码做文档。</li>
</ul>
<blockquote>
<p>Info: 你的代码注释可以写成 Markdown 格式。<code v-pre>yiisoft/yii2-apidoc</code> 扩展为你提供了一个从你的 代码注释生成漂亮的 API 文档。</p>
</blockquote>
<blockquote>
<p>Info: 虽然不做要求，我们还是建议你的扩展遵循某个编码规范。你可以参照 <a href="https://github.com/yiisoft/yii2/blob/master/docs/internals/core-code-style.md" target="_blank" rel="noopener noreferrer">核心框架代码风格</a>。</p>
</blockquote>
<h2 id="核心扩展" tabindex="-1"><a class="header-anchor" href="#核心扩展"><span>核心扩展</span></a></h2>
<p>Yii 提供了下列核心扩展，由 Yii 开发团队开发和维护。这些扩展全都在 <a href="https://packagist.org/" target="_blank" rel="noopener noreferrer">Packagist</a> 中注册，并像 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/structure-extensions.md#using-extensions" target="_blank" rel="noopener noreferrer">Using Extensions</a> 章节描述的那样容易安装。</p>
<ul>
<li><a href="https://github.com/yiisoft/yii2-apidoc" target="_blank" rel="noopener noreferrer">yiisoft/yii2-apidoc</a>：提供了一个可扩展的、高效的 API 文档生成器。核心框架的 API 文档也是用它生成的。</li>
<li><a href="https://github.com/yiisoft/yii2-authclient" target="_blank" rel="noopener noreferrer">yiisoft/yii2-authclient</a>：提供了一套常用的认证客户端，例如 Facebook OAuth2 客户端、GitHub OAuth2 客户端。</li>
<li><a href="https://github.com/yiisoft/yii2-bootstrap" target="_blank" rel="noopener noreferrer">yiisoft/yii2-bootstrap</a>：提供了一套挂件，封装了 <a href="http://getbootstrap.com/" target="_blank" rel="noopener noreferrer">Bootstrap</a> 的组件和插件。</li>
<li><a href="https://github.com/yiisoft/yii2-debug" target="_blank" rel="noopener noreferrer">yiisoft/yii2-debug</a>：提供了对 Yii 应用的调试支持。当使用该扩展是，在每个页面的底部将显示一个调试工具条。该扩展还提供了一个独立的页面，以显示更详细的调试信息。</li>
<li><a href="https://github.com/yiisoft/yii2-elasticsearch" target="_blank" rel="noopener noreferrer">yiisoft/yii2-elasticsearch</a>： 提供对 <a href="https://www.elastic.co/" target="_blank" rel="noopener noreferrer">Elasticsearch</a> 的使用支持。它包含基本的查询/搜索支持，并实现了 <a href="https://github.com/yiisoft/yii2/blob/master/docs/guide-zh-CN/db-active-record.md" target="_blank" rel="noopener noreferrer">Active Record</a> 模式让你可以将活动记录 存储在 Elasticsearch 中。</li>
<li><a href="https://github.com/yiisoft/yii2-faker" target="_blank" rel="noopener noreferrer">yiisoft/yii2-faker</a>： 提供了使用 <a href="https://github.com/fzaninotto/Faker" target="_blank" rel="noopener noreferrer">Faker</a> 的支持，为你生成模拟数据。</li>
<li><a href="https://github.com/yiisoft/yii2-gii" target="_blank" rel="noopener noreferrer">yiisoft/yii2-gii</a>： 提供了一个基于页面的代码生成器，具有高可扩展性，并能用来快速生成模型、 表单、模块、CRUD 等。</li>
<li><a href="https://github.com/yiisoft/yii2-httpclient" target="_blank" rel="noopener noreferrer">yiisoft/yii2-httpclient</a>： 提供 HTTP 客户端。</li>
<li><a href="https://github.com/yiisoft/yii2-imagine" target="_blank" rel="noopener noreferrer">yiisoft/yii2-imagine</a>： 提供了基于 <a href="http://imagine.readthedocs.org/" target="_blank" rel="noopener noreferrer">Imagine</a> 的常用图像处理功能。</li>
<li><a href="https://github.com/yiisoft/yii2-jui" target="_blank" rel="noopener noreferrer">yiisoft/yii2-jui</a>： 提供了一套封装 <a href="http://jqueryui.com/" target="_blank" rel="noopener noreferrer">JQuery UI</a> 的挂件以及它们的交互。</li>
<li><a href="https://github.com/yiisoft/yii2-mongodb" target="_blank" rel="noopener noreferrer">yiisoft/yii2-mongodb</a>： 提供了对 <a href="https://www.mongodb.com/" target="_blank" rel="noopener noreferrer">MongoDB</a> 的使用支持。它包含基本 的查询、活动记录、数据迁移、缓存、代码生成等特性。</li>
<li><a href="https://www.yiiframework.com/extension/yiisoft/yii2-queue" target="_blank" rel="noopener noreferrer">yiisoft/yii2-queue</a>： 通过队列异步提供运行任务的支持。 它支持基于 DB，Redis，RabbitMQ，AMQP，Beanstalk 和 Gearman 的队列。</li>
<li><a href="https://github.com/yiisoft/yii2-redis" target="_blank" rel="noopener noreferrer">yiisoft/yii2-redis</a>： 提供了对 <a href="http://redis.io/" target="_blank" rel="noopener noreferrer">redis</a> 的使用支持。它包含基本的 查询、活动记录、缓存等特性。</li>
<li><a href="https://www.yiiframework.com/extension/yiisoft/yii2-shell" target="_blank" rel="noopener noreferrer">yiisoft/yii2-shell</a>： 提供基于 <a href="http://psysh.org/" target="_blank" rel="noopener noreferrer">psysh</a> 的交互式 shell。</li>
<li><a href="https://github.com/yiisoft/yii2-smarty" target="_blank" rel="noopener noreferrer">yiisoft/yii2-smarty</a>： 提供了一个基于 <a href="http://www.smarty.net/" target="_blank" rel="noopener noreferrer">Smarty</a> 的模板引擎。</li>
<li><a href="https://github.com/yiisoft/yii2-sphinx" target="_blank" rel="noopener noreferrer">yiisoft/yii2-sphinx</a>： 提供了对 <a href="http://sphinxsearch.com/" target="_blank" rel="noopener noreferrer">Sphinx</a> 的使用支持。它包含基本的 查询、活动记录、代码生成等特性。</li>
<li><a href="https://github.com/yiisoft/yii2-swiftmailer" target="_blank" rel="noopener noreferrer">yiisoft/yii2-swiftmailer</a>： 提供了基于 <a href="http://swiftmailer.org/" target="_blank" rel="noopener noreferrer">swiftmailer</a> 的邮件发送功能。</li>
<li><a href="https://github.com/yiisoft/yii2-twig" target="_blank" rel="noopener noreferrer">yiisoft/yii2-twig</a>： 提供了一个基于 <a href="https://twig.symfony.com/" target="_blank" rel="noopener noreferrer">Twig</a> 的模板引擎。</li>
</ul>
<p>以下官方扩展适用于 Yii 2.1 及以上版本。 您不需要为 Yii 2.0 安装它们，因为它们包含在核心框架中。</p>
<ul>
<li><a href="https://www.yiiframework.com/extension/yiisoft/yii2-captcha" target="_blank" rel="noopener noreferrer">yiisoft/yii2-captcha</a>： 提供 CAPTCHA。</li>
<li><a href="https://www.yiiframework.com/extension/yiisoft/yii2-jquery" target="_blank" rel="noopener noreferrer">yiisoft/yii2-jquery</a>： 为 <a href="https://jquery.com/" target="_blank" rel="noopener noreferrer">jQuery</a> 提供支持。</li>
<li><a href="https://www.yiiframework.com/extension/yiisoft/yii2-maskedinput" target="_blank" rel="noopener noreferrer">yiisoft/yii2-maskedinput</a>： 提供基于 <a href="http://robinherbots.github.io/Inputmask/" target="_blank" rel="noopener noreferrer">jQuery Input Mask plugin</a> 的格式化输入小部件。</li>
<li><a href="https://www.yiiframework.com/extension/yiisoft/yii2-mssql" target="_blank" rel="noopener noreferrer">yiisoft/yii2-mssql</a>： 提供对使用 <a href="https://www.microsoft.com/sql-server/" target="_blank" rel="noopener noreferrer">MSSQL</a> 的支持。</li>
<li><a href="https://www.yiiframework.com/extension/yiisoft/yii2-oracle" target="_blank" rel="noopener noreferrer">yiisoft/yii2-oracle</a>： 提供对使用 <a href="https://www.oracle.com/" target="_blank" rel="noopener noreferrer">Oracle</a> 的支持。</li>
<li><a href="https://www.yiiframework.com/extension/yiisoft/yii2-rest" target="_blank" rel="noopener noreferrer">yiisoft/yii2-rest</a>： 提供对 REST API 的支持。</li>
</ul>
<blockquote>
<p>💖喜欢本文档的，欢迎点赞、收藏、留言或转发，谢谢支持！<br>
作者邮箱：zhuzixian520@126.com，github地址：<a href="https://github.com/zhuzixian520" target="_blank" rel="noopener noreferrer">github.com/zhuzixian520</a></p>
</blockquote>
</div></template>


